/**
 * Copyright (C) 2004-2011 Jive Software. All rights reserved.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package tic.tac.toe;

import java.awt.Color;
import java.awt.Font;
import java.util.HashSet;

import javax.swing.ImageIcon;
import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;

import org.jivesoftware.resource.Res;
import org.jivesoftware.smack.SmackException;
import org.jivesoftware.smack.filter.StanzaIdFilter;
import org.jivesoftware.smack.iqrequest.AbstractIqRequestHandler;
import org.jivesoftware.smack.iqrequest.IQRequestHandler;
import org.jivesoftware.smack.packet.IQ;
import org.jivesoftware.smack.packet.Presence;
import org.jivesoftware.smack.packet.StanzaError;
import org.jivesoftware.smack.provider.ProviderManager;
import org.jivesoftware.spark.SparkManager;
import org.jivesoftware.spark.plugin.Plugin;
import org.jivesoftware.spark.ui.ChatRoom;
import org.jivesoftware.spark.ui.ChatRoomButton;
import org.jivesoftware.spark.ui.ChatRoomListener;
import org.jivesoftware.spark.ui.ChatRoomListenerAdapter;
import org.jivesoftware.spark.ui.rooms.ChatRoomImpl;
import org.jivesoftware.spark.util.log.Log;
import org.jxmpp.jid.EntityBareJid;
import org.jxmpp.jid.EntityFullJid;
import org.jxmpp.jid.parts.Localpart;
import tic.tac.toe.packet.GameOfferPacket;
import tic.tac.toe.packet.InvalidMove;
import tic.tac.toe.packet.MovePacket;
import tic.tac.toe.ui.GamePanel;


/**
 * Tic Tac Toe plugin for Spark
 * 
 * @author wolf.posdorfer
 * @version 16.06.2011
 * 
 */
public class TicTacToePlugin implements Plugin {

    private ChatRoomListener _chatRoomListener;
    private IQRequestHandler _gameOfferHandler;
    
    private HashSet<EntityBareJid> _currentInvitations;
       
    private ImageIcon buttonimg;
    
    

    @Override
    public void initialize() {
	ClassLoader cl = getClass().getClassLoader();
	buttonimg = new ImageIcon(cl.getResource("ttt.button.png"));
	_currentInvitations = new HashSet<>();

	ProviderManager.addIQProvider(GameOfferPacket.ELEMENT_NAME, GameOfferPacket.NAMESPACE, new GameOfferPacket.Provider() );
	ProviderManager.addExtensionProvider(MovePacket.ELEMENT_NAME, MovePacket.NAMESPACE, new MovePacket.Provider() );
	ProviderManager.addExtensionProvider(InvalidMove.ELEMENT_NAME, InvalidMove.NAMESPACE, new InvalidMove.Provider() );

	// Add IQ listener to listen for incoming game invitations.
    _gameOfferHandler = new AbstractIqRequestHandler(
        GameOfferPacket.ELEMENT_NAME,
        GameOfferPacket.NAMESPACE,
        IQ.Type.get,
        IQRequestHandler.Mode.async)
    {
        public IQ handleIQRequest(IQ request) {
            showInvitationAlert((GameOfferPacket) request);
            return null;
        }
    };

    SparkManager.getConnection().registerIQRequestHandler(_gameOfferHandler);
	addButtonToToolBar();

    }
    /**
     * Add the TTT-Button to every opening Chatroom
     * and create Listeners for it
     */
    private void addButtonToToolBar() {
	

	_chatRoomListener = new ChatRoomListenerAdapter() {
	    @Override
	    public void chatRoomOpened(final ChatRoom room) {
		
		if(!(room instanceof ChatRoomImpl))
		{
		    // Don't do anything if this is not a 1on1-Chat
		    return;
		}
		
		final ChatRoomButton sendGameButton = new ChatRoomButton(buttonimg);
		room.getToolBar().addChatRoomButton(sendGameButton);


            sendGameButton.addActionListener(e -> {
                
                //If the opponent is offline, then you should not start the game
                if(((ChatRoomImpl) room).getPresence().getType().equals(Presence.Type.unavailable)){
                    return;
                }
                final EntityFullJid opponentJID = ((ChatRoomImpl) room).getJID();
                if (_currentInvitations.contains(opponentJID.asBareJid())) {
                    return;
                }

                final GameOfferPacket offer = new GameOfferPacket();
                offer.setTo(opponentJID);
                offer.setType(IQ.Type.get);

                _currentInvitations.add(opponentJID.asEntityBareJid());
                room.getTranscriptWindow().insertCustomText
                    (TTTRes.getString("ttt.request.sent"), false, false, Color.BLUE);
                try {
                    SparkManager.getConnection().sendStanza(offer);
                } catch (SmackException.NotConnectedException | InterruptedException e1) {
                    Log.warning("Unable to send offer to " + opponentJID, e1);
                }

                SparkManager.getConnection().addAsyncStanzaListener(
                    stanza -> {
                        if (stanza.getError() != null) {
                            room.getTranscriptWindow().insertCustomText
                                (TTTRes.getString("ttt.request.decline"), false, false, Color.RED);
                            _currentInvitations.remove(opponentJID.asBareJid());
                            return;
                        }

                        GameOfferPacket answer = (GameOfferPacket) stanza;
                        answer.setStartingPlayer(offer.isStartingPlayer());
                        answer.setGameID(offer.getGameID());
                        if (answer.getType() == IQ.Type.result) {
                            // ACCEPT
                            _currentInvitations.remove(opponentJID.asBareJid());

                            room.getTranscriptWindow().insertCustomText
                                (TTTRes.getString("ttt.request.accept"), false, false, Color.BLUE);

                            createTTTWindow(answer, opponentJID);
                        } else {
                            // DECLINE
                            room.getTranscriptWindow().insertCustomText
                                (TTTRes.getString("ttt.request.decline"), false, false, Color.RED);
                            _currentInvitations.remove(opponentJID.asBareJid());
                        }
                    }, new StanzaIdFilter(offer));
                    // TODO: Just filtering by stanza id is insure, should use Smack's IQ send-response mechanisms.
            });
	    }

	    @Override
	    public void chatRoomClosed(ChatRoom room) {
		if (room instanceof ChatRoomImpl) {
		    ChatRoomImpl cri = (ChatRoomImpl) room;
		    _currentInvitations.remove(cri.getParticipantJID());
		}
	    }
	};
	

	SparkManager.getChatManager().addChatRoomListener(_chatRoomListener);

    }

    @Override
    public void shutdown() {

	_currentInvitations.clear();
	SparkManager.getChatManager().removeChatRoomListener(_chatRoomListener);
        SparkManager.getConnection().unregisterIQRequestHandler(_gameOfferHandler);
    }

    @Override
    public boolean canShutDown() {
	return false;
    }

    @Override
    public void uninstall() {

    }

    /**
     * insert the Invitation Dialog into the Chat
     * 
     * @param invitation
     */
    private void showInvitationAlert(final GameOfferPacket invitation) {
        invitation.setType(IQ.Type.result);
        invitation.setTo(invitation.getFrom());

        final ChatRoom room = SparkManager.getChatManager().getChatRoom(invitation.getFrom().asEntityBareJidOrThrow());

        Localpart name = invitation.getFrom().getLocalpartOrThrow();

        final JPanel panel = new JPanel();
        JLabel text = new JLabel(TTTRes.getString("ttt.game.request", name));
        JLabel game = new JLabel(TTTRes.getString("ttt.game.name"));
        game.setFont(new Font("Dialog", Font.BOLD, 24));
        game.setForeground(Color.RED);
        JButton accept = new JButton(Res.getString("button.accept").replace("&", ""));
        JButton decline = new JButton(Res.getString("button.decline").replace("&", ""));
        panel.add(text);
        panel.add(game);
        panel.add(accept);
        panel.add(decline);

        room.getTranscriptWindow().addComponent(panel);

        accept.addActionListener(e -> {
            try {
                SparkManager.getConnection().sendStanza(invitation);
            } catch (SmackException.NotConnectedException | InterruptedException e1) {
                Log.warning("Unable to send invitation accept to " + invitation.getTo(), e1);
            }
            invitation.setStartingPlayer(!invitation.isStartingPlayer());
            createTTTWindow(invitation, invitation.getTo().asEntityFullJidOrThrow());
            panel.remove(3);
            panel.remove(2);
            panel.repaint();
            panel.revalidate();
        });

        decline.addActionListener(e -> {
            invitation.setType(IQ.Type.error);
            invitation.setError(StanzaError.getBuilder().setCondition(StanzaError.Condition.undefined_condition).setDescriptiveEnText("User declined your request."));
            try {
                SparkManager.getConnection().sendStanza(invitation);
            } catch (SmackException.NotConnectedException | InterruptedException e1) {
                Log.warning("Unable to send invitation decline to " + invitation.getTo(), e1);
            }
            panel.remove(3);
            panel.remove(2);
            panel.repaint();
            panel.revalidate();
        });
    }

    /**
     * Creates The TicTacToe Window and starts the Game
     * @param gop
     * @param opponentJID
     */
    private void createTTTWindow(GameOfferPacket gop, EntityFullJid opponentJID) {
	
	Localpart name = opponentJID.getLocalpart();
	
	// tictactoe versus ${name}
	JFrame f = new JFrame(TTTRes.getString("ttt.window.title", TTTRes.getString("ttt.game.name"),name.toString() ));
	
	f.setIconImage(buttonimg.getImage());
	GamePanel gp = new GamePanel(SparkManager.getConnection(),
		gop.getGameID(), gop.isStartingPlayer(), opponentJID,f);
	f.add(gp);
	f.pack();
	f.setLocationRelativeTo(SparkManager.getChatManager().getChatContainer());
	f.setVisible(true);
	
    }

}
